package cn.always.xiajia.framework.web.core.handler;

import org.springframework.core.MethodParameter;
import org.springframework.http.MediaType;
import org.springframework.http.server.ServerHttpRequest;
import org.springframework.http.server.ServerHttpResponse;
import org.springframework.http.server.ServletServerHttpRequest;
import org.springframework.web.bind.annotation.ControllerAdvice;
import org.springframework.web.servlet.mvc.method.annotation.ResponseBodyAdvice;

import cn.always.xiajia.framework.common.entity.AjaxResult;

/**
 * 全局响应结果（ResponseBody）处理器
 *
 * 不同于在网上看到的很多文章，会选择自动将 Controller 返回结果包上 {@link CommonResult}， 在 onemall 中，是
 * Controller 在返回时，主动自己包上 {@link CommonResult}。 原因是，GlobalResponseBodyHandler
 * 本质上是 AOP，它不应该改变 Controller 返回的数据结构
 *
 * 目前，GlobalResponseBodyHandler 的主要作用是，记录 Controller 的返回结果， 方便
 * {@link cn.iocoder.yudao.framework.apilog.core.filter.ApiAccessLogFilter}
 * 记录访问日志
 */
@ControllerAdvice
public class GlobalResponseBodyHandler implements ResponseBodyAdvice {

	@Override
	@SuppressWarnings("NullableProblems") // 避免 IDEA 警告
	public boolean supports(MethodParameter returnType, Class converterType) {
		if (returnType.getMethod() == null) {
			return false;
		}
		// 只拦截返回结果为 CommonResult 类型
		return returnType.getMethod().getReturnType() == AjaxResult.class;
	}

	@Override
	@SuppressWarnings("NullableProblems") // 避免 IDEA 警告
	public Object beforeBodyWrite(Object body, MethodParameter returnType, MediaType selectedContentType, Class selectedConverterType,
			ServerHttpRequest request, ServerHttpResponse response) {
		((ServletServerHttpRequest) request).getServletRequest().setAttribute("ajax_result", (AjaxResult) body);
		return body;
	}

}
